## CSE 141L Milestone 2

Christopher Edwards, A16855415

Keisuke Hirano, A16974148

Oscar Zheng, A16880333

## **Academic Integrity**

Your work will not be graded unless the signatures of all members of the group are present beneath the honor code.

To uphold academic integrity, students shall:

- Complete and submit academic work that is their own and that is an honest and fair representation of their knowledge and abilities at the time of submission.
- Know and follow the standards of CSE 141L and UCSD.

Please sign (type) your name(s) below the following statement:

I pledge to be fair to my classmates and instructors by completing all of my academic work with integrity. This means that I will respect the standards set by the instructor and institution, be responsible for the consequences of my choices, honestly represent my knowledge and abilities, and be a community member that others can trust to do the right thing even when no one is watching. I will always put learning before grades, and integrity before performance. I pledge to excel with integrity.

Oscar Zheng Keisuke Hirano Christopher Edwards

#### 0. Team

Oscar Zheng, Keisuke Hirano, Christopher Edwards

#### 1. Introduction

TODO. Name your architecture. What is your overall philosophy? What specific goals did you strive to achieve? Can you classify your machine in any of the standard ways (e.g., stack machine, accumulator, register-register/load-store, register-memory)? If so, which? If not, devise a name for your class of machine. Word limit: 200 words.

Our general philosophy was creating an architecture that was capable of implementing algorithms with simple instructions that deal with computation primarily at register level without storing much in main memory. In our design, we specifically targeted programs 1 and 2 and neglected whether the ISA has the capability to solve program 3. We designed our ISA to be very RISC-oriented, in that it offers limited arithmetic instructions and not much flexibility with immediate values. It relies on significant register manipulation and moving values around. Our machine follows an accumulator design, where one register is the dedicated destination register and the first register operand. We also have a

register to hold the single flag that the ALU sets. We included two more, unusual instructions, movf and movt, in order to make data maneuverability easier in the accumulator register.

### 2. Architectural Overview

TODO. This must be in picture form. What are the major building blocks you expect your processor to be made up of? You must have data memory in your architecture. (Example of MIPS: <a href="https://www.researchgate.net/figure/The-MIPS-architecture\_fig1\_251924531">https://www.researchgate.net/figure/The-MIPS-architecture\_fig1\_251924531</a>)



# 3. Machine Specification

### Instruction formats

Two example rows have been filled for you. When you submit, do not include the example types. Add rows as necessary. In your submission, please delete this paragraph.

| TYPE | FORMAT                          | CORRESPONDING INSTRUCTIONS              |  |
|------|---------------------------------|-----------------------------------------|--|
| R    | 4 bit opcode, 4 bit reg         | add, xor, sub, ld, str, movf, movt, cmp |  |
| J    | 4 bit opcode, 5-bit jump offset | jmp, jc, jnn, jlt, jeq                  |  |
| I    | 4 bit opcode, 5 bit immediate   | movi, Isl, rsl                          |  |

#### Operations

An example row has been filled for you. When you submit, do not include the example type. In the name column, be sure to also add the definition of what the example actually does. For example, "Isl = logical shift left" would be an appropriate value to put in the name column. In the bit breakdown column, add in parenthesis what specific values the bits should be in order. X indicates that it will be specified by the programmer's instruction itself (i.e. specifying registers). In the example column, give an example of an "assembly language" instruction in your machine, then translate it into machine code. Add rows as necessary. In your submission, please delete this paragraph.

| NAME                          | TYPE | BIT BREAKDOWN                                           | EXAMPLE                                                                     | NOTES                                                                                                         |
|-------------------------------|------|---------------------------------------------------------|-----------------------------------------------------------------------------|---------------------------------------------------------------------------------------------------------------|
| jmp = jump<br>unconditionally | J    | 4 bit opcode<br>(0000), 5 bit index<br>of look-up table | Before: PC_LUT[3] = 0x05 PC = 0xff jmp #3 After: PC = 0x05                  | jump to the absolute address given by the value in the PC LUT at the index that's provided in the instruction |
| jc = jump if carry out        | J    | 4 bit opcode<br>(0001), 5 bit index<br>of look-up table | Before: PC_LUT[3] = 0x05<br>C = 1<br>PC = 0xff<br>jc #3<br>After: PC = 0x05 | jump if the C flag (Carry out) is set                                                                         |
| jlt = jump less<br>than       | J    | 4 bit opcode<br>(0010), 5 bit index<br>of look-up table | Before: PC_LUT[3] = 0x05<br>N = 1<br>PC = 0xff<br>jlt #3                    | jump if the N flag (Negative) is set                                                                          |

|                                      |   |                                                         | After: PC = 0x05                                                                  |                                                                                                                                                 |
|--------------------------------------|---|---------------------------------------------------------|-----------------------------------------------------------------------------------|-------------------------------------------------------------------------------------------------------------------------------------------------|
| jeq = jump if<br>equal to            | J | 4 bit opcode (0011),<br>5 bit index of<br>look-up table | Before: PC_LUT[3] = 0x05 Z = 1 PC = 0xff jeq #3 After: PC = 0x05                  | jump if the Z flag (zero) is set                                                                                                                |
| jnn = jump not<br>negative<br>number | J | 4 bit opcode<br>(0100), 5 bit index<br>of look-up table | Before: PC_LUT[3] = 0x05<br>N = 0<br>PC = 0xff<br>jnn #3<br>After: PC = 0x05      | jump if the N flag is reset                                                                                                                     |
| add                                  | R | 4 bit opcode<br>(0101), 4 bit<br>register (XXXX)        | Before: R0 = 0b0000_0001<br>R1 = 0b1000_0000<br>add R1<br>After: R0 = 0b1000_0001 | adds the accumulator register and given register and stores in accumulator. Sets the Carry out flag if the addition resulted in a carry out bit |
| xor                                  | R | 4 bit opcode (0110),<br>4 bit register<br>(XXXX)        | Before: R0 = 0b0000_0001<br>R1 = 0b1000_0000<br>xor R1                            | xors the accumulator register and given register and stores in accumulator register                                                             |

|                         |   |                                                  | After: R0 = 0b1000_0001                                                                    |                                                                                                                 |
|-------------------------|---|--------------------------------------------------|--------------------------------------------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------|
| sub                     | R | 4 bit opcode (0111),<br>4 bit register<br>(XXXX) | Before: R0 = 0b0001_0000<br>R1 = 0b0001_0001<br>sub R1<br>After: R0 = 0b1111_1111<br>N = 1 | subtracts the given register from the accumulator and stores it in the accumulator. Sets the N (negative) flag. |
| ld                      | R | 4 bit opcode<br>(1000), 4 bit<br>register (XXXX) | Before: Memory[0x05] = 0x10<br>R1 = 0x05<br>Idr R1<br>After: R0 = 0x10                     | load from the address given in the operand register from memory and store value in the accumulator register     |
| str                     | R | 4 bit code (1001), 4<br>bit register (XXXX)      | Before: R0 = 0x10<br>R1 = 0x05<br>str R1<br>After: Memory[0x05] = 0x10                     | store the value in the accumulator register to the address in memory given by the operand register              |
| movf = move<br>from reg | R | 4 bit opcode<br>(1010), 4 bit<br>register (XXXX) | Before: R0 = 0x05<br>movf R1<br>After: R1 = 0x05                                           | moves information from accumulator register to the operand register                                             |
| movt = move<br>to reg   | R | 4 bit opcode (1011),<br>4 bit register<br>(XXXX) | Before: R0 = 0x00<br>R1 = 0x05                                                             | moves the value in the operand register to the accumulator register                                             |

|                              |   |                                                    | movt R1  After: R0 = 0x05                                                       |                                                                                                                                                                          |
|------------------------------|---|----------------------------------------------------|---------------------------------------------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| стр                          | R | 4 bit opcode (1100),<br>4 bit register<br>(XXXX)   | Before: R0 = 0x05<br>R1 = 0x05<br>cmp R1<br>After: Z = 1                        | subtracts the operand register from the accumulator register, sets the N (negative) flag if the result is 0                                                              |
| IsI = left shift<br>logical  | I | 4 bit opcode (1101),<br>5 bit immediate<br>(XXXXX) | Before: R0 = 0b0100_0001<br>C = 0<br>ls1 #2<br>After: R0 = 0b0000_0100<br>C = 1 | logical left shift the value in the accumulator register by a number of iterations given by the immediate value operand. Inputs the last shifted-out bit into the C flag |
| rsl = right shift<br>logical |   | 4 bit opcode (1111),<br>5 bit immediate<br>(XXXXX) | Before: R0 = 0b0100_0001<br>C = 0<br>rsl #2<br>After: R0 = 0b0001_0000          | logical right shift the value in the accumulator register by a number of iterations given by the immediate value operand.                                                |
| movi = move<br>immediate     | I | 4 bit opcode (1110),<br>5 bit immediate<br>(XXXXX) | Before: R0 = 0x00 movi #13 After: R0 = 0x0D                                     | moves immediate to the accumulator register                                                                                                                              |

#### **Internal Operands**

TODO. How many registers are supported? Is there anything special about any of the registers (e.g. constant, accumulator), or all of them general purpose?

The number of registers we support are 16 registers. Even though there are 5 bits, because of the limitations that were given, we think that this is an efficient way of using the registers. These registers would be accumulator registers.

#### Control Flow (branches)

TODO. What types of branches are supported? How are the target addresses calculated? What is the maximum branch distance supported? How do you accommodate large jumps?

We support 5 types of branches which are: jump unconditionally, jump when equal, jump when less than, jump when carry out, and jump when non negative. There is only one flag which is set in different scenarios; the flag being set signifies different things dependent on the instruction before it. The cmp instruction would set the flag to true if both registers were equal, sub would set to true if the operator register is greater than the accumulator, and IsI would set the flag to the bit that was shifted out.

The addresses for jumps are stored in a hard coded look up table before the program is run and we jump by referencing the register that contains the index of the address in the lookup table. The maximum branch distance is assumed to be infinite as the lookup table can hold very large numbers.

If there is a jump longer than the available bytes for the lookup tables, then we would do something like a compound jump where we would jump to one address which jumps again to another address.

#### **Addressing Modes**

TODO. What memory addressing modes are supported, e.g. direct, indirect? How are addresses calculated? Give examples. We are using both direct and indirect. In some cases, we have a register holding the address being used such as the when jumping. We use indirect addresses for the R-type instructions. For example, add R7 would sum the values stored in R0 and R7, and store it in R0. We also use indirect addressing for jump instructions. For example, a jump instruction would be *jlt #4* which would mean for the program to jump to the address at index 4 of the lookup table. Our I-type instructions are the only ones that use direct addressing. We use we use direct addresses such as *movi #22* which would let the program know to move the number 22 into the register.

### 4. Programmer's Model [Lite]

TODO. 4.1 How should a programmer think about how your machine operates? Provide a description of the general strategy a programmer should use to write programs with your machine. For example, one could say that the programmer should prioritize loading in the necessary values from memory into as many registers as possible, then perform calculations. Another approach could be loading and writing to memory in between every calculation step. Word limit: 200 words.

A programmer using our architecture should generally think about storing all the values they need into a lookup table before running the program. We only offer 16 registers so they should be moderately conservative about the amount of data they have to store while running the program. These registers are only 8 bits wide as well, giving yet another reason to store some of the values needed before the program is run. Because the ISA is an accumulator, they should be prepared to think about moving many values in and out of registers to be able to manipulate multiple values at a time.

TODO. 4.2 Can we copy the instructions/operation from MIPS or ARM ISA? If no, explain why not? How did you overcome this or how do you deal with this in your current design? Word limit: 100 words.

We are unable to copy the instructions from MIPS or ARM ISA because of the difference in the format of the instructions. In MIPS and ARM, their instructions use up to three register operands, one destination register, and two operating registers. In our instructions, we are using an accumulator register which means for our instructions, we are only using one register which all eventually lead to the accumulator register. This does not mean that ARM or MIPS functions and codes would not work in our program, but it means that there would be more instructions because we would need to move information out of the accumulator and into another register or vice versa.

TODO. 4.3 Will your ALU be used for non-arithmetic instructions (e.g., MIPS or ARM-like memory address pointer calculations, PC relative branch computations, etc.)? If so, how does that complicate your design?

No, our ALU is only used for our R-type (arithmetic) instructions. We do not allow for addressing offsets; any memory accesses must have the absolute address desired to be accessed stored in a register.

## 5. Program Implementation

An example Pseudocode and Assembly Code has been filled out for you. When you submit, please delete the example along with this paragraph.

#### Example Pseudocode

```
// Program 1
int i, j, xorResult;
int min = 16, max=0; // MOV min=16, max=0
for (i = 0; i < 31; i++) \{ // MOV i = 0; JLT i < 31; ADD i + 1 \}
     for (j = i+1; j<32; j++) { // MOV j = i+1; JLT j<32; ADD j+1}
          xorResult = arr[i] ^ arr[j]; // LD arr[i]; LD arr[j]; XOR
           int c = 0; // MOV c = 0
           int count1s = 0; // MOV count1s = 0
           while (c < 16) {
                xorResult << 1; // LSL 1; shift MSB to Carry bit
                if (carryflag = 1) { // JC - jump if Carry flag is set
                     count1s++; // ADD count1s + 1
                c++; // ADD c + 1 }
                // here countls holds Hamming Distance bt arr[i] and arr[j]
                if (max < count1s) { // JLT</pre>
                     max = count1s; // MOV
                if (countls < min) { // JLT
                     min = count1s; // MOV
```

# Example Assembly Code

Assembly code provided in Github

#### ChangeLog from Milestone 1

- 1. Changed the number of bits to specify the operand register from 5 to 4. We realized that the assignment specifies we can only have 16 registers.
- 2. Added a lookup table for the program counter in order to store jump addresses. We were using registers to begin with, but this way, jumps are limited to 255 addresses away. With the lookup table, we are not bounded by how far the PC can jump.
- 3. Added the cmp and jeq instruction to set the flag if the two registers are equal, and to jump in the case that they are equal.
- 4. Added the str instruction because we realized the testbench read the memory values to see our answer.
- 5. Added a jmp instruction to unconditionally jump to the target when called.
- 6. Changed from 3 flags to 1 flag. We will have 1 flag which can represent different scenarios depending on what the instruction was that set the flag. cmp sets the flag if the result of subtracting Rs from R0 is 0. sub sets the flag if the resulting value from subtracting Rs from R0 is negative. 1s1 sets the flag if the bit that was shifted out last was 1.
- 7. Dedicated R15 to be the status register that will hold the value of the flag. We realized we needed a place to easily store and access the flag.

#### ChangeLog from Milestone 2

- 1. Added a right shift logical rsl operation.
- 2. Changed the status register from R15 to R3.

## ChangeLog Final

- 1. Changed the add instruction to set the flag if there is a carry out bit.
- 2. Decided that the jump instructions must be immediately after instruction that is meant to set the flag that determines whether the jump is taken. There cannot be other instructions between.